summaryrefslogtreecommitdiff
path: root/app/(tabs)/notifications/[id].tsx
diff options
context:
space:
mode:
Diffstat (limited to 'app/(tabs)/notifications/[id].tsx')
-rw-r--r--app/(tabs)/notifications/[id].tsx301
1 files changed, 301 insertions, 0 deletions
diff --git a/app/(tabs)/notifications/[id].tsx b/app/(tabs)/notifications/[id].tsx
new file mode 100644
index 0000000..e2d39ba
--- /dev/null
+++ b/app/(tabs)/notifications/[id].tsx
@@ -0,0 +1,301 @@
+import {
+ Alert,
+ Platform,
+ StyleSheet,
+} from 'react-native';
+import ParallaxScrollView from '@/components/ParallaxScrollView';
+import { ThemedText } from '@/components/ThemedText';
+import { ThemedView } from '@/components/ThemedView';
+import React, { useState, useCallback, useEffect, useRef } from 'react';
+import AsyncStorage from '@react-native-async-storage/async-storage';
+import { useFocusEffect } from '@react-navigation/native';
+import { Link, router, useLocalSearchParams } from 'expo-router';
+import MapView, { Marker, Polygon, Region } from 'react-native-maps';
+import { Ionicons } from '@expo/vector-icons';
+import { useColorScheme } from '@/hooks/useColorScheme.web';
+import { Colors } from '@/constants/Colors';
+
+interface AlertData {
+ id: string;
+ userId: string;
+ createdAt: string;
+ area: string;
+ extendedArea: string;
+ level: string;
+}
+
+interface PositionData {
+ id: string;
+ userId: string;
+ createdAt: string;
+ latitude: number;
+ longitude: number;
+ movingActivity: string;
+}
+
+interface NotificationData {
+ id: string;
+ alert: AlertData;
+ position: PositionData;
+ seen: boolean;
+ createdAt: string;
+}
+
+interface PolygonCoordinates {
+ latitude: number;
+ longitude: number;
+}
+
+export default function NotificationIdScreen() {
+ const [token, setToken] = useState<string>('');
+ const [userId, setUserId] = useState<string>('');
+ const [notification, setNotification] = useState<NotificationData | null>(null);
+ const [region, setRegion] = useState<Region>({
+ latitude: 44.49738301084014,
+ longitude: 11.356121722966094,
+ latitudeDelta: 0.03,
+ longitudeDelta: 0.03,
+ });
+ const [coordinates, setCoordinates] = useState({ latitude: 44.49738301084014, longitude: 11.356121722966094 });
+ const [polygon, setPolygon] = useState<PolygonCoordinates[]>([]);
+ const [extendedPolygon, setExtendedPolygon] = useState<PolygonCoordinates[]>([]);
+ const mapRef = useRef<MapView | null>(null);
+
+ const checkAuth = useCallback(async () => {
+ const storedToken =
+ Platform.OS === 'web'
+ ? localStorage.getItem('token')
+ : await AsyncStorage.getItem('token');
+ const storedUserId =
+ Platform.OS === 'web'
+ ? localStorage.getItem('userId')
+ : await AsyncStorage.getItem('userId');
+
+ setToken(storedToken || '');
+ setUserId(storedUserId || '');
+
+ if (!storedToken || !storedUserId) {
+ Alert.alert(
+ 'Login required',
+ 'You must log in to the system if you want to see notifications list',
+ [
+ {
+ text: 'Ok',
+ onPress: () => router.push('/'),
+ },
+ ]
+ );
+ }
+ }, []);
+
+ const updateSeenStatus = async (id: string) => {
+ const response = await fetch(`${process.env.EXPO_PUBLIC_API_URL}/graphql`, {
+ method: 'POST',
+ headers: {
+ 'Authorization': `Bearer ${token}`,
+ 'Content-Type': 'application/json',
+ },
+ body: JSON.stringify({
+ query: `
+ mutation NotificationUpdate($input: NotificationUpdateInput!) {
+ notificationUpdate(input: $input) { id seen }
+ }`,
+ variables: {
+ input: {
+ id,
+ seen: true,
+ },
+ },
+ }),
+ });
+
+ const data = await response.json();
+ if (data.errors) {
+ console.error(`On editing notification "${id}": ${JSON.stringify(data)}`);
+ }
+ };
+
+ const fetchNotification = useCallback(async (id: string) => {
+ if (!token || !userId) return;
+
+ try {
+ const response = await fetch(
+ `${process.env.EXPO_PUBLIC_API_URL}/graphql`,
+ {
+ method: 'POST',
+ headers: {
+ Authorization: `Bearer ${token}`,
+ 'Content-Type': 'application/json',
+ },
+ body: JSON.stringify({
+ query: `{ notifications(id: ${id}) {
+ id,
+ alert { id, userId, createdAt, area, extendedArea, level, reachedUsers },
+ position {id, userId, createdAt, latitude, longitude, movingActivity},
+ seen,
+ createdAt
+ } }`,
+ }),
+ }
+ );
+
+ const data = await response.json();
+
+ if (data.errors) {
+ Alert.alert('Error', 'Error fetching data');
+ } else if (data.data.notifications && data.data.notifications.length > 0) {
+ const notificationData = data.data.notifications[0];
+ const coordinatesString = notificationData.alert.area.substring(9, notificationData.alert.area.length - 2);
+ const coordinates = coordinatesString
+ .split(',')
+ .map((coord: string) => coord.trim().split(' '))
+ .map((pair: string[]) => ({
+ longitude: parseFloat(pair[0]),
+ latitude: parseFloat(pair[1]),
+ }));
+
+ const extendedCoordinatesString = notificationData.alert.extendedArea.substring(9, notificationData.alert.extendedArea.length - 2);
+ const extendedCoordinates = extendedCoordinatesString
+ .split(',')
+ .map((coord: string) => coord.trim().split(' '))
+ .map((pair: string[]) => ({
+ longitude: parseFloat(pair[0]),
+ latitude: parseFloat(pair[1]),
+ }));
+
+ setCoordinates({ latitude: notificationData.position.latitude, longitude: notificationData.position.longitude });
+ setNotification(notificationData);
+ setPolygon(coordinates);
+ setExtendedPolygon(extendedCoordinates);
+ setRegion({
+ latitude: coordinates[0]?.latitude || region.latitude,
+ longitude: coordinates[0]?.longitude || region.longitude,
+ latitudeDelta: 0.05,
+ longitudeDelta: 0.05,
+ });
+
+ return notificationData;
+ } else {
+ Alert.alert('Error', 'No data found');
+ router.push('/notifications/index');
+ }
+ } catch (err) {
+ console.error('Fetch Map Data Error:', err);
+ }
+
+ }, [token, userId, region.latitude, region.longitude]);
+
+ useFocusEffect(
+ useCallback(() => {
+ checkAuth();
+ }, [checkAuth])
+ );
+
+ const { id } = useLocalSearchParams();
+
+ useEffect(() => {
+ if (typeof id === 'string') {
+ fetchNotification(id).then((n) => {
+ if (n && !n.seen) {
+ updateSeenStatus(n.id);
+ }
+ })
+ }
+ }, [id, fetchNotification]);
+
+ useEffect(() => {
+ if (mapRef.current && region) {
+ mapRef.current.animateToRegion(region, 1000);
+ }
+ }, [region]);
+
+ const formatDate = (timestamp: string) => {
+ const date = new Date(parseInt(timestamp, 10) * 1000);
+ return `${date.toDateString()} ${date.getHours()}:${(date.getMinutes() < 10 ? '0' : '') + date.getMinutes()}`;
+ };
+
+ const theme = useColorScheme() ?? 'light';
+
+ return (
+ <ParallaxScrollView token={token} userId={userId}>
+ {notification === null ? (
+ <ThemedView>
+ <ThemedText>Loading...</ThemedText>
+ </ThemedView>
+ ) : (
+ <>
+ <ThemedView style={styles.dateRow}>
+ <Ionicons
+ name="chevron-back-outline"
+ size={18}
+ color="#0a7ea4"
+ style={styles.icon}
+ />
+ <Link href="/notifications">
+ <ThemedText type="link">Notifications list</ThemedText>
+ </Link>
+ </ThemedView>
+ <ThemedView>
+ <MapView
+ ref={mapRef}
+ initialRegion={region}
+ style={styles.map}
+ >
+ <Marker coordinate={coordinates} title="You were here" isPreselected />
+ <Polygon
+ coordinates={polygon}
+ strokeColor="#c0392b"
+ fillColor="rgba(192, 57, 43, 0.4)"
+ />
+ <Polygon
+ coordinates={extendedPolygon}
+ strokeColor="#c0392b"
+ fillColor="rgba(192, 57, 43, 0.4)"
+ />
+ </MapView>
+ </ThemedView>
+ <ThemedView style={styles.dateRow}>
+ <Ionicons
+ name="calendar-outline"
+ size={18}
+ color={theme === 'light' ? Colors.light.text : Colors.dark.text}
+ style={styles.icon}
+ />
+ <ThemedText>Notified: {formatDate(notification.createdAt)}</ThemedText>
+ </ThemedView>
+ <ThemedView style={styles.dateRow}>
+ <Ionicons
+ name="calendar-outline"
+ size={18}
+ color={theme === 'light' ? Colors.light.text : Colors.dark.text}
+ style={styles.icon}
+ />
+ <ThemedText>Alerted: {formatDate(notification.alert.createdAt)}</ThemedText>
+ </ThemedView>
+ <ThemedView style={styles.dateRow}>
+ <Ionicons
+ name="alert-circle-outline"
+ size={18}
+ color={theme === 'light' ? Colors.light.text : Colors.dark.text}
+ style={styles.icon}
+ />
+ <ThemedText>Level of alert: {notification.alert.level}</ThemedText>
+ </ThemedView>
+ </>
+ )}
+ </ParallaxScrollView>
+ );
+}
+
+const styles = StyleSheet.create({
+ map: {
+ height: 400,
+ },
+ dateRow: {
+ flexDirection: 'row',
+ alignItems: 'center',
+ },
+ icon: {
+ marginRight: 8,
+ },
+});